// ==UserScript== // @name Bilibili Dynamic Block // @namespace xiaohuohumax/userscripts/bilibili-dynamic-block // @version 1.0.1 // @author xiaohuohumax // @description Bilibili 动态拦截 // @license MIT // @icon https://static.hdslb.com/mobile/img/512.png // @source https://github.com/xiaohuohumax/userscripts.git // @match https://t.bilibili.com/* // @match https://space.bilibili.com/* // @require https://unpkg.com/sweetalert@2.1.2/dist/sweetalert.min.js // @grant GM_addStyle // @grant GM_addValueChangeListener // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_setValue // @noframes // ==/UserScript== (e=>{if(typeof GM_addStyle=="function"){GM_addStyle(e);return}const o=document.createElement("style");o.textContent=e,document.head.append(o)})(" .swal-overlay,.swal-overlay input{color:#000000a6}.swal-overlay .swal-button--success{background-color:#a3dd82}.swal-overlay .swal-button--success:hover{background-color:#98d973}.swal-overlay .swal-title{padding-top:10px;padding-bottom:10px}.swal-overlay hr{border-color:#00000024;margin:10px 1px 5px}.swal-overlay .add-rule-container{display:flex;margin:6px 0}.add-rule-container input{border-radius:6px 0 0 6px}.add-rule-container button{flex-shrink:0;border-radius:0 6px 6px 0}.rules-container{min-height:200px;max-height:220px;overflow-y:auto}.rules-container .empty{padding:8px;font-size:14px}.rules-container .rules-item{display:flex;margin:6px 0;position:relative}.rules-container .rules-item input{border-radius:6px}.swal-overlay .rules-item .close-item{top:50%;transform:translateY(-50%);right:10px;position:absolute;background-image:url();background-position:50%;background-repeat:no-repeat;background-size:cover;height:22px;width:22px;cursor:pointer}.swal-overlay .rules-item .close-item:hover{transform:translateY(-50%) scale(1.125)}#bilibili-dynamic-block-stat{position:fixed;bottom:3px;right:3px;font-size:10px;z-index:999;cursor:pointer}#bilibili-dynamic-block-stat:hover{font-weight:900} "); (function (swal) { 'use strict'; var __defProp = Object.defineProperty; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value); var _GM_addValueChangeListener = /* @__PURE__ */ (() => typeof GM_addValueChangeListener != "undefined" ? GM_addValueChangeListener : void 0)(); var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)(); var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)(); var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)(); const debounce = ({ delay }, func) => { let timer = void 0; let active = true; const debounced = (...args) => { if (active) { clearTimeout(timer); timer = setTimeout(() => { active && func(...args); timer = void 0; }, delay); } else { func(...args); } }; debounced.isPending = () => { return timer !== void 0; }; debounced.cancel = () => { active = false; }; debounced.flush = (...args) => func(...args); return debounced; }; const ID = "bilibili-dynamic-block"; const VERSION = "1.0.1"; const LAST_VERSION = 1; class Store { constructor() { __publicField(this, "config", null); __publicField(this, "ID", `${ID}-config`); __publicField(this, "listeners", []); this.loadConfig(); _GM_addValueChangeListener(this.ID, (_key, _oldValue, newValue, remote) => { if (remote) { this.config = this.configFormat(newValue); this.listeners.forEach((listener) => listener(this.config)); } }); } loadConfig() { const config = _GM_getValue(this.ID, void 0); this.config = this.configFormat(config); !config && this.saveConfig(); console.log("加载配置:", this.config); } saveConfig() { _GM_setValue(this.ID, this.config); this.listeners.forEach((listener) => listener(this.config)); } addConfigChangeListener(listener) { this.listeners.push(listener); } configFormat(data) { const config = { version: LAST_VERSION, blockRules: [], showStat: false }; if (!data) { return config; } if (data.version === 0) { return config; } return Object.assign(config, data); } addBlockRule(rule) { if (!rule) { return; } for (const r of Array.isArray(rule) ? rule : [rule]) { const rTrim = r.trim(); if (rTrim === "" || this.config.blockRules.includes(rTrim)) { continue; } this.config.blockRules.unshift(rTrim); } this.saveConfig(); } deleteBlockRule(rule) { if (!rule) { return false; } const index = this.config.blockRules.indexOf(rule); if (index === -1) { return false; } this.config.blockRules.splice(index, 1); this.saveConfig(); return true; } updateBlockRule(oldRule, newRule) { const index = this.config.blockRules.indexOf(oldRule); if (index === -1) { return; } this.config.blockRules[index] = newRule; this.saveConfig(); } clearBlockRules() { this.config.blockRules = []; this.saveConfig(); } get blockRules() { return this.config.blockRules; } get showStat() { return this.config.showStat; } async exportConfig() { try { const data = JSON.stringify(this.config, null, 2); const blob = new Blob([data], { type: "text/json" }); const url = URL.createObjectURL(blob); const a = document.createElement("a"); a.href = url; a.download = `${ID}-config.json`; a.click(); URL.revokeObjectURL(url); return true; } catch { return false; } } async importConfig(file) { return new Promise((resolve) => { const reader = new FileReader(); reader.readAsText(file); reader.onload = () => { try { const oldRules = this.config.blockRules; this.config = this.configFormat(JSON.parse(reader.result)); this.addBlockRule(oldRules); resolve(true); } catch { resolve(false); } }; }); } toggleShowStat() { this.config.showStat = !this.config.showStat; this.saveConfig(); } } class View { constructor(store2) { __publicField(this, "statInfo", ""); __publicField(this, "statElement", null); __publicField(this, "renderConfig", async () => { const element = document.createElement("div"); const emptyContent = '